package com.atguigu.gulimall.search.thread;

import com.sun.org.apache.xpath.internal.functions.FuncTrue;

import java.util.concurrent.*;

public class ThreadTest {
    public static ExecutorService executor = Executors.newFixedThreadPool(10);

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        System.out.println("main----start-----");
//        CompletableFuture.runAsync(()->{
//            System.out.println("当前线程:"+Thread.currentThread().getId());
//            int i = 10/2;
//            System.out.println("运行结果:"+i);
//        },executor);
        /**
         * 方法完成后的感知
         */
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(()->{
//            System.out.println("当前线程:"+Thread.currentThread().getId());
//            int i = 10/10;
//            System.out.println("运行结果:"+i);
//            return i;
//            },executor).whenComplete((res,exception)->{
//                //虽然能得到异常信息,但是没法修改返回数据
//            System.out.println("异步任务完成了。。。。结果是:"+res+";异常是:"+exception);
//        }).exceptionally(throwable -> {
//            //可以感知异常,同时返回默认值
//            return 10;
//        });
        /**
         * 方法完成后的处理
         */
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(()->{
//            System.out.println("当前线程:"+Thread.currentThread().getId());
//            int i = 10/10;
//            System.out.println("运行结果:"+i);
//            return i;
//        },executor).handle((res,throwable)->{
//            if(res != null){
//                return res*2;
//            }
//            if(throwable!=null){
//                return 0;
//            }
//            return 0;
//        });
        /**
         * 线程串行化
         * 1)、thenRun: 不能获取到上一步的执行结果,无返回值
         * .thenRunAsync(()->{
         *     System.out.println("任务2启动了...");
         * },executor);
         * 2)、thenAcceptAsync;能接收上一步的结果，但是无返回值
         * 3)、thenApplyAsync;能接收上一步骤的结果，有返回值
         */
//      CompletableFuture<String> future = CompletableFuture.supplyAsync(()->{
//            System.out.println("当前线程:"+Thread.currentThread().getId());
//            int i = 10/10;
//            System.out.println("运行结果:"+i);
//            return i;
//            },executor).thenApplyAsync(res -> {
//          System.out.println("任务2执行了。。。。"+res);
//          return "Hello"+res;
//      },executor);
        /**
         * 两个都完成
         */
//        CompletableFuture<Object> future01 = CompletableFuture.supplyAsync(()->{
//            System.out.println("任务1线程:"+Thread.currentThread().getId());
//            int i = 10/2;
//            System.out.println("任务1结束");
//            return i;
//        },executor);
//         CompletableFuture<Object> future02 = CompletableFuture.supplyAsync(()->{
//             System.out.println("任务2线程:"+ Thread.currentThread().getId());
//             System.out.println("任务2线程结束");
//             return "hello";
//        },executor);
        //任务组合
//         future01.runAfterBothAsync(future02,()->{
//             System.out.println("任务3");
//         },executor);
        //R apply(T t,U u);
//         String a = future.get();
//        System.out.println("main----end-----"+a);
//        future01.thenAcceptBothAsync(future02,(f1,f2)->{
//            System.out.println("任务3开始...之前的执行结果是"+f1+"...."+f2);
//        },executor);
//        CompletableFuture<String> future = future01.thenCombineAsync(future02,(f1,f2)->{
//            return f1+ f2 +"->hello";
//        },executor);
//        String s = future.get();
//        System.out.println("最后返回的结果是"+s);
        /**
         * 两个任务,只要有一个完成,我们就执行任务3
         * runAfterEitherAsync:不感知结果,自己没有返回值
         * acceptEitherAsync：感知结果,自己没有返回值
         */
//        future01.runAfterEitherAsync(future02,()->{
//            System.out.println("任务3开始执行");
//        },executor);
//        future01.acceptEitherAsync(future02,(res)->{
////            System.out.println("任务3开始执行...之前的结果是:"+ res);
////        },executor);
//        CompletableFuture<String> future = future01.applyToEitherAsync(future02,(res)->{
//            System.out.println("任务3开始执行...之前的结果是:"+ res);
//            return res.toString()+"哈哈";
//            },executor);
        CompletableFuture<String> futureImg = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询图片的信息");
            return "hello.jpg";
        },executor);
        CompletableFuture<String> futureAttr = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询商品的属性");
            return "黑色+256G";
        },executor);
        CompletableFuture<String> futureDesc = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询商品介绍");
            return "华为";
        },executor);
        CompletableFuture<Void> allOf = CompletableFuture.allOf(futureImg,futureAttr,futureDesc);
        allOf.get(); //join()等待所有的线程完成  anyOf 任何一个
        System.out.println("main----end----");
    }
    public static void thread(String[] args) {
        System.out.println("main----start-----");
        /**
         * 1) 、继承Thread
         * //          Thread thread = new Thread01();
         * //        thread.start();
         * 2) 、实现Runnable接口
         * //        Runable01 runable01 = new Runable01();
         * //        Thread thread = new Thread(runable01);
         * //        thread.start();
         * 3) 、实现Callable接口 + FutureTask (可以拿到返回的结果,可以处理异常)
         * //        FutureTask<Integer> futureTask = new FutureTask<>(new Callable01());
         * //        new Thread(futureTask).start();
         * //        等待整个线程执行完成,获取返回的结果
         * //        Integer integer = futureTask.get();
         * 4) 、 线程池[ExecutorService]
         *          给线程池直接提交任务 service.execute(new Runable01());
         *          1、创建:
         *              1) 、Executors
         *              2）、 new ThreadPoolExecutor
         *区别 ：
         *      1、2不能带到返回值,3可以得到返回值
         *      1，2，3 都不能控制资源
         *      4 可以控制资源,性能稳定
         */
        //以上三种启动线程的方式都不用,将所有的多线程异步任务都叫个线程池执行
//        new Thread(()-> System.out.println("hello")).start();
        //当前系统中池只有一两个,每个异步任务,提交给线程让他自己去执行就行
//        service.execute(new Runable01());
        /**
         * 七大参数
         * corePoolSize:[5] 核心线程数[一直存在除非（allowCoreThreadTimeOut）]; 线程池： 创建号以后就准备就绪的线程数量，就等待来接收异步任务的去执行
         *              5 个 Thread thread = new Thread(); thread.start();
         * maximumPoolSize:[200] 最大线程数量 ，控制资源
         * keepAliveTime: 存活时间.如果当前的线程数量大于core数量。
         *              释放空闲的线程(maximumPoolSize-corePoolSize) . 只要线程空闲大于指定的keepAliveTime;
         * unit: 时间单位
         * BlockingQueue<Runnable> workQueue: 阻塞队列.如果任务有很多，就会将目前的任务放在队列里面
         *                  只要有空闲线程,就会去队列里面去除新的任务继续执行
         * threadFactory: 线程创建的工厂
         * rejectedExecutionHandler  handler : 如果队列满了,按照我们指定的拒绝策略拒绝执行任务
         *
         * 工作顺序:
         * 1)、线程池创建,准备好core数量的核心线程,准备接收任务
         * 1.1、core 满了,就将再进入的任务放入阻塞队列中,空闲的core就会自己去阻塞队列获取任务执行
         * 1.2、阻塞队列满了,就执行开启新的线程执行,最大只能开到max指定的数量
         * 1.3、max满了就用RejectedExecutionHandler拒绝任务
         * 1.4、max都执行完成,有很多空闲,在指定的时间keepAliveTime以后,释放max-core这些线程
         * new LinkedBlockingDeque<>() ： 默认是Integer的最大值,内存不够
         * 一个线程池 core 7 : max 20 queue: 50 ,100 并发进来怎么分配的
         * 7个会立即执行,50个会进入队列,在开13个进行执行,剩下的就会使用拒绝策略
         * 如果不想抛弃还得执行, CallerRunsPolicy
         */
        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                5,
                200,
                10,
                TimeUnit.SECONDS,
                new LinkedBlockingDeque<>(100000),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.AbortPolicy()
        );
        System.out.println("main----end-----");

    }
    public static class Thread01 extends Thread{
        @Override
        public void run() {
            System.out.println("当前线程:"+ Thread.currentThread().getId());
            int i = 10/2;
            System.out.println("运行结果:"+i);
        }
    }
    public static class Runable01 implements Runnable{

        @Override
        public void run() {
            System.out.println("当前线程:"+ Thread.currentThread().getId());
            int i = 10/2;
            System.out.println("运行结果:"+i);
        }
    }
    public static class Callable01 implements Callable<Integer> {

        @Override
        public Integer call() throws Exception {
            System.out.println("当前线程:"+ Thread.currentThread().getId());
            int i = 10/2;
            System.out.println("运行结果:"+i);
            return i;
        }
    }

}
